

Course Name: Embedded Systems Design

**Course Number**: 16:332:579

**Assignment**: Lab 3– Where no clock has gone before

Course Instructor: Prof. Phillip Southard

**Date Submitted**: 3/28/2019

**Submitted by**: Sanjana Devaraj (NetID: sd1049)

# **Contents**

- 1 Purpose
- 2 Lab Assignment 1: UART Transmitter
  - 2.1 Theory of Operation
  - 2.2 Design
  - 2.3 Test
  - 2.4 Implementation
- 3 Lab Assignment 2: Sender
  - 3.1 Theory of Operation
  - 3.2 Design
  - 3.3 Implementation
- 4 Lab Assignment 3: Sender Top-Level
  - 4.1 Theory of Operation
  - 4.2 Design
  - 4.3 Implementation
- 5 Discussion
  - 5.1 Observations
  - 5.2 Problem Areas

### 1 Purpose

The main purpose of this lab was to introduce the concept of a Finite State Machine (FSM) and use it to create a sequential design, a Universal Asynchronous Receiver Transmitter (UART). By using a FSM to control the behavior of a circuit in a certain sequence, an output is produced that was read by the computer and displayed in a terminal emulator.

The UART takes bytes of data and transmits the individual bits in a sequential fashion (serial manner). Serial transmission of digital data (bits) through a single wire or other medium is much more effective than parallel transmission through multiple wires because it minimizes capacitive coupling of wires. The UART transmits data asynchronously. Asynchronous transmission allows data to be transmitted without the sender having to send a clock signal to the receiver. In this case, the sender and receiver must agree on timing parameters (Baud Rate) prior transmission and special bits are added to each word to synchronize the sending and receiving units (which are optional and are used depending on user requirements). Common serial communication uses the protocol shown below:



Figure 1: UART Frame Waveform

The lab is comprised of two tasks. The first task consisted of two sub-sections, design of the transmitter part of the UART that sends data using the protocol shown in Figure 1 and testing the top level UART design consisting of UART receiver, UART transmitter sub-modules with the provided UART testbench. The second consisted of two sub-sections, design of a sender module which sends data using a FSM and implementation phase consisted of instantiating the UART, sender, modified clock divider that produces a 115200 Hz output, and a pair of debouncing logic designs in the final sender top level design.

#### Prelab Task:

The FSM state diagram for a device that takes as input an 8-bit data packet and transmits it using the protocol in Figure 1 is shown below:



### 2 Lab Assignment 1: Design of UART transmitter

### 2.1 Theory of Operation

The UART transmitter transmits sends data using the protocol in Figure 1 with specific behavior. The signal ready is '1' when it is in idle state. All internal registers are cleared and the transition happens to idle state when reset is '1'. The data is stored into a register and the sequence of sending data begins on the rising clock edge when send is '1' and enable is '1'.

### 2.2 Design

Listing 1: sender\_tx.vhd

```
library IEEE;
use IEEE.std logic 1164.all;
use IEEE.numeric std.all;
entity uart_tx is
port (clk , en , send , rst : in std logic ;
char : in std logic vector (7 downto 0);
ready , tx : out std logic);
end uart_tx ;
architecture Behavioral of uart_tx is
type state is (idle, start, data);
signal curr : state := idle;
signal d : std logic vector(7 downto 0) := (others => '0');
signal d_count : std logic vector(3 downto 0);
process(clk)
begin
if rising edge(clk) then
   if rst = '1' then
     d <= (others => '0');
     ready <= '1';
      tx <='1';
      curr <= idle;
      d count <= (others => '0');
   end if;
   if en = '1' then
     case curr is
     when idle =>
     ready <= '1';
```

```
tx <= '1';
      if send = '1' then
         d <= char;
         curr <= start;
      else
         ready <= '1';
        tx <= '1';
         curr <= idle;
      end if;
      when start =>
      ready <='0';
       tx <= '0';
       curr <= data;
       when data =>
       if unsigned(d_count) < 8 then
          tx <= d(to integer(unsigned(d count)));</pre>
          d_count <= std logic vector(unsigned(d_count) + 1);</pre>
          curr <= data;
       else
           ready <= '1';
           tx <= '1';
           d_count <= (others => '0');
           curr <= idle;
       end if;
       end case;
    end if;
end if;
end process;
end Behavioral;
```

### 2.3 Test

The top level UART design consisting of UART receiver, UART transmitter sub-modules is tested with the provided UART testbench.

### Simulations results:



Figure 2: UART Simulation



Figure 3: UART Simulation, Zoomed in

# 2.4 Implementation

# 2.4.1 UART Transmitter Design

# 2.4.1.1 Elaboration Schematic:



Figure 4: UART\_tx RTL Schematic

# 2.4.1.2 Synthesis Schematic:



Figure 5: UART\_tx Synthesis Schematic

### 2.4.1.3 Post-Synthesis Utilization Table:



Figure 6: UART\_tx Post-Synthesis Utilization Table

### 2.4.1.4 On-chip Power Graph:



Figure 7: UART\_tx On-chip Power Graph

# 2.4.2 Top level UART Design

### 2.4.2.1 Elaboration Schematic:



Figure 8: UART RTL Schematic

# 2.4.2.2 Synthesis Schematic:



Figure 9: UART Synthesis Schematic

### 2.4.3.3 Post-Synthesis Utilization Table:

| Utilization |          | Post-Synthesis   Post-Implementation |           |             |  |  |
|-------------|----------|--------------------------------------|-----------|-------------|--|--|
|             | Graph    |                                      |           |             |  |  |
|             | Resource | Estimation                           | Available | Utilization |  |  |
|             | LUT      | 27                                   | 17600     | 0.15        |  |  |
|             | FF       | 43                                   | 35200     | 0.12        |  |  |
|             | IO       | 24                                   | 100       | 24.00       |  |  |
|             | BUFG     | 1                                    | 32        | 3.13        |  |  |

Figure 10: UART Post-Synthesis Utilization Table

### 2.4.3.4 On-chip Power Graph:



Figure 11: UART On-chip Power Graph

# 3 Lab Assignment 2: Sender

### 3.1 Theory of Operation

The sender module transmits my NetID (sd1049) which is a n-long array of 8-bit vectors where n=6 using a FSM having a specific behavior. The sender module takes as input a reset, a clock, a clock enable, a button and a 'ready' signal and outputs a 'send' signal and an 8-bit 'char'. The FSM is initialized to an idle state and changes on the rising edge of the clock when enable is '1'

according to the following mentioned conditions. When a reset signal is asserted, all outputs and counter i are cleared and transition happens to idle state. When ready is '1', button is '1' and i <6, 'send' is set to '1' and NETID(i) is stored on 'char', counter i is incremented, and transition happens to a 'busyA' state. The counter i to reset to '0' and state remains in idle state when 'ready' is '1' and button is '1' but i = 6. After entering 'busyA' state, transition happens to 'busyB' state. After entering 'busyB' state, 'send' value is changed to '0' and transition happens to 'busyC' state. If 'ready' is '1' and button is '0' then transition happens to idle state, otherwise it remains in 'busyC' state.

### 3.2 Design

Listing 2: sender.vhd

```
21 | library IEEE;
22 ;
    use IEEE.std logic 1164.all;
23 use IEEE.numeric std.all;
24
25 entity sender is
    port(rst, clk, en, btn, rdy : in std logic;
27 | send : out std logic;
28 char : out std logic vector(7 downto 0));
29 @ end sender;
31 - architecture Behavioral of sender is
32
33 type str is array (0 to 5) of std logic vector(7 downto 0);
34 | signal NETID : str := (X"73", X"64", X"31", X"30", X"34", X"39");
35 type state is (idle, busyA, busyB, busyC);
    signal curr : state := idle;
36
    signal i : std logic vector(4 downto 0);
37 :
38
39 begin
40 FSM: process(clk, en)
41
       begin
42 :
43 E
           if rising edge(clk) and en = '1' then
44 🖨
               if rst = 'l' then
45 !
                    char <= "000000000";
46
                    i <= "000000";
47
                    curr <= idle;
                    send <= '0';
48
49 🖨
                end if:
50 □
           case curr is
51 🖯
            when idle =>
52 🖯
           if rdy = 'l' and btn = 'l' then
53 🖯
              if unsigned(i) < 6 then
54 :
                 send <= '1';
```

```
55 1
                   char <= netid(to_integer(unsigned(i)));</pre>
56
                   i <= std logic vector(unsigned(i) + 1);
57
                   curr <= busyA;
58
                else
59
                   i <= "000000";
60
                   curr <= idle;
61 🖨
                end if;
62 🖨
             end if;
63
64 🖯
            when busyA =>
65 🖨
            curr <= busyB;
66
67 🖨
            when busyB =>
             send <= '0';
68 ;
69 🖨
             curr <= busyC;
70 ;
71 🖯
            when busyC =>
72 🖯
            if rdy = '1' and btn = '0' then
73 :
                curr <= idle;
74 🖨
             end if;
75 🖨
          end case;
76 🖨
         end if;
77 end process;
78 end Behavioral;
```

## 3.3 Implementation

### 3.3.1 Elaboration Schematic:



Figure 12: Sender RTL Schematic

# 3.3.2 Synthesis Schematic:



Figure 13: Sender Synthesis Schematic

# 3.3.3 Post-Synthesis Utilization Table:

| Utilization |          | Post-Synthesis   Post-Implementation |           |             |  |
|-------------|----------|--------------------------------------|-----------|-------------|--|
|             |          | Graph   Table                        |           |             |  |
|             | Resource | Estimation                           | Available | Utilization |  |
|             | LUT      | 17                                   | 17600     | 0.10        |  |
|             | FF       | 13                                   | 35200     | 0.04        |  |
|             | 10       | 14                                   | 100       | 14.00       |  |
|             | BUFG     | 1                                    | 32        | 3.13        |  |

Figure 14: Sender Post-Synthesis Utilization Table

### 3.3.4 On-chip Power Graph:

#### Summary On-Chip Power Power analysis from Implemented netlist, Activity derived from constraints files, simulation files or Dynamic: 0.001 W (2%)vectorless analysis. Total On-Chip Power: 0.093 W 38% Clocks: 0.001 W (38%) **Design Power Budget:** Not Specified Signals: <0.001 W (4%)98% **Power Budget Margin:** N/A Logic: <0.001 W (5%)53% **Junction Temperature:** 26.1°C I/O: 0.001 W (53%) Thermal Margin: 58.9°C (5.0 W) Device Static: 0.091 W (98%) Effective 9JA: 11.5°C/W Power supplied to off-chip devices: 0 W Confidence level: Launch Power Constraint Advisor to find and fix invalid switching activity

Figure 15: Sender On-chip Power Graph

# 4 Lab Assignment 3: Sender Top level

# 4.1 Theory of Operation

The designed UART, sender, modified clock divider that produces a 115200 Hz output, and a pair of debouncing logic designs are instantiated using structural VHDL modeling in this final sender top level design.

### 4.2 Design

#### 4.2.1 Clock divider

The clock divider circuit produces the clock divided signal as output with a frequency of 115200 Hz which is the specified Baud Rate for this assignment, the rate at which the Transmitter sends data and the Receiver receives data.

The input clock frequency is 125 MHz.

 $\Rightarrow$  125\* 10<sup>6</sup>/1085 = 115207.37 Hz (approximately equal to the required band rate 115200 Hz)

### Listing 3: clock\_div.vhd

```
22 | library IEEE;
23  use IEEE.std_logic_1164.all;
24 use IEEE.numeric std.all;
25 !
26 - entity clk_div is
27 port (clk : in std logic;
28 div : out std logic);
29 end clk div;
30 ;
31 architecture clk_ckt of clk_div is
32 | signal counter : std_logic_vector (25 downto 0) := (others => '0');
33
34 | begin
35 - process(clk)
36 begin
37 \ominus if rising_edge(clk) then
38 if (unsigned(counter) < 1085) then
39 :
      div <= '0';
counter <= std_logic_vector( unsigned(counter) + 1 );</pre>
40
41
      else
      counter <= (others => '0');
       div <= 'l';
43
44 ← end if;
45 end if;
46 end process;
48 end clk ckt;
```

### 4.2.2 Sender Top Level Design

Listing 4: top\_level.vhd

```
22 library IEEE;
23 | use IEEE.std logic 1164.all;
24
25 - entity top_level is
26 port (TXD, clk : in std logic;
27 btn : in std logic vector(1 downto 0);
28 CTS, RTS, RXD: out std logic);
29 end top_level;
31 architecture top level ckt of top level is
33 | signal dbnce: std logic vector(1 downto 0);
34 ! signal div, send, ready, tx : std logic;
35 | signal char: std logic vector(7 downto 0);
36
37 - component debounce is
38 port ( clk: in std logic;
39 | btn: in std logic;
40 dbnc: out std logic);
41 @ end component;
43 - component clk div
44 | port ( clk : in std logic;
45 div : out std logic);
46 end component;
47
48 - component sender is
49 port ( rst, clk, en, btn, rdy : in std logic;
50 send : out std logic;
51 | char : out std logic vector(7 downto 0));
52 end component;
```

```
54 - component uart is
 55 ; port(clk, en, send, rx, rst : in std logic;
 56 charSend : in std logic vector (7 downto 0);
 57 ; ready, tx, newChar : out std logic;
 58 charRec : out std logic vector (7 downto 0));
 59 end component;
 60
 61 begin
 62
 63 CTS <= '0';
 64 : RTS <= '0';
 65
 66 - ul: debounce
 67 | port map(clk => clk,
 68 | btn => btn(0),
 69 @ dbnc => dbnce(0));
 70
 71 □ u2: debounce
72 | port map(clk => clk,
73 | btn => btn(1),
 74 @ dbnc => dbnce(1));
75
 76 U3: clk div
77 port map(clk => clk,
78 @ div => div);
79 :
80 🗇 u4: sender
81 : port map ( btn => dbnce(1),
82 | clk => clk,
83 en => div,
84
    rdy => ready,
   rst => dbnce(0),
86 ; char => char,
87 ( send => send);
89 🖯 u5: uart
90 port map ( charSend => char,
    clk => clk,
91
92 en => div,
93 : rst => dbnce(0),
94 rx => TXD,
95 send => send,
    ready => ready,
96
97 \( \tau \) tx => RXD);
98
99 end top_level_ckt;
```

# 4.3 Implementation

### 4.3.1 Elaboration Schematic:



Figure 16: Sender Top Level RTL Schematic

# 4.3.2 Synthesis Schematic:



Figure 17: Sender Top Level Synthesis Schematic

### 4.3.3 Post-Synthesis Utilization Table:

| Utilization |          | Post-Synthesis   Post-Implementation |           |             |  |
|-------------|----------|--------------------------------------|-----------|-------------|--|
|             |          | Graph   Table                        |           |             |  |
|             | Resource | Estimation                           | Available | Utilization |  |
|             | LUT      | 119                                  | 17600     | 0.68        |  |
|             | FF       | 100                                  | 35200     | 0.28        |  |
|             | IO       | 7                                    | 100       | 7.00        |  |
|             | BUFG     | 1                                    | 32        | 3.13        |  |

Figure 18: Sender Top Level Post-Synthesis Utilization Table

### 4.3.4 On-chip Power Graph:



Figure 19: Sender Top Level On-chip Power Graph

#### 4.3.5 Constraints File

The constraints (XDC) file was changed to map the appropriate signals on the board to the final sender top level design. The lines uncommented in the XDC file are shown in the below figures. A operating frequency of clock 125 MHz needs to sent to the Zybo board, 2 push buttons need to be enabled to observe the transmission of dataon the serial terminal. Button 0 is used for reset and button 1 is used for sending data in a sequential manner. The UART Pmod is connected to the Pmod header JB on the board. The lines of code corresponding to Pmod header JB was uncommented so that pins T20, U20, V20, W20 are enabled for the signals RTS, RXD, TXD, CTS respectively.

### Listing 5: ZYBO\_Master.xdc

```
1 ## This file is a general .xdc for the ZYBO Rev B board
 2  ## To use it in a project:
3 · ## - uncomment the lines corresponding to used pins
 4 ! ## - rename the used signals according to the project
6
7 : ##Clock signal
8 set property -dict { PACKAGE PIN L16 IOSTANDARD LVCMOS33 } [get ports { clk }]; #IO L11P T1 SRCC 35 Sch=sysclk
9 | create clock -add -name sys_clk_pin -period 8.00 -waveform {0 4} [get_ports { clk }];
10
11 !
12 ##Switches
14 ' #set property -dict ( PACKAGE PIN P15 IOSTANDARD LVCMOS33 ) [get ports { sw[1] }]; #IO L24P T3 34 Sch=SW1
15 : #set property -dict ( PACKAGE PIN W13 IOSTANDARD LVCMOS33 ) [get ports { sw[2] }]; #IO L4N TO 34 Sch=SW2
17 1
18 !
19 : ##Buttons
20 set property -dict { PACKAGE PIN R18 IOSTANDARD LVCMOS33 } [get ports { btn[0] }]; #IO L20N T3 34 Sch=BTN0
21 set property -dict { PACKAGE_PIN P16 IOSTANDARD LVCMOS33 } [get ports { btn[1] }]; #IO L24N T3 34 Sch=BTN1
22 #set property -dict { PACKAGE PIN V16 IOSTANDARD LVCMOS33 } [get ports { btn[2] }]; #IO L18P T2 34 Sch=BTN2
23 | #set property -dict { PACKAGE PIN V16 | IOSTANDARD LVCMOS33 } [get ports { btn[3] }]; #IO L7P T1 34 Sch=BTN3
79 i
80 | ##Pmod Header JB
81 set property -dict { PACKAGE_PIN T20 IOSTANDARD LVCMOS33 } [get ports { RTS }]; #IO L15P T2 DQS 34 Sch=JB1 p
82 | set property -dict { PACKAGE PIN U20
                                      IOSTANDARD LVCMOS33 } [get ports { RXD }]; #IO L15N T2 DQS 34 Sch=JB1 N
83 | set property -dict { PACKAGE PIN V20
                                      IOSTANDARD LVCMOS33 } [get ports { TXD }]; #IO L16P T2 34 Sch=JB2 P
84 | set property -dict { PACKAGE PIN W20
                                      IOSTANDARD LVCMOS33 } [get ports { CTS }]; #IO L16N T2 34 Sch=JB2 N
85 | #set property -dict { PACKAGE PIN Y18
                                       IOSTANDARD LVCMOS33 } [get ports { jb p[2] }]; #IO L17P T2 34 Sch=JB3 P
86 | #set property -dict ( PACKAGE PIN Y19
                                       IOSTANDARD LVCMOS33 } [get ports { jb n[2] }]; #IO L17N T2 34 Sch=JB3 N
87 | #set property -dict ( PACKAGE PIN W18
                                       IOSTANDARD LVCMOS33 } [get ports { jb p[3] }]; #IO L22P T3 34 Sch=JB4 P
88 #set property -dict ( PACKAGE PIN W19 IOSTANDARD LVCMOS33 ) [get ports ( jb n[3] )]; #IO L22N T3 34 Sch=JB4 N
89 !
```

### 4.3.6 Results:

The serial terminal 'Termite' settings are shown below:



Figure 20: Termite Settings

When button 1 is pressed 6 times consecutively, the output is as shown in the below figure:



Figure 21: Output

#### 5 Discussion

#### 5.1 Observations

This lab assignment helped to understand the concept of FSM, serial communication, the working of UART in depth. My concepts of FSM are much clear now after using FSM in a practical application such as a hardware device like UART.

#### 5.2 Problem areas

I spent long time in the first task getting the UART transmitter part correct. I am getting better at debugging by running the testbench code to observe the simulation results and tracing back to where I went wrong. My problem was that when I ran the UART module with the testbench the first byte which was received was 'ff' instead of '48'. I initially though it was a problem with the testbench because the time period for which the ready and tx signal were high was longer than the other times these signals were high as shown in the figure below because of which the received byte was 'ff'. This problem was corrected by initializing the data counter variable 'd\_count' to zero when reset is high.



Figure 22: Simulation of UART with incorrect first byte

The sender FSM design and sender top level design were fairly simple.